Newer
Older
BlackoutClient / Assets / Best HTTP / Source / SecureProtocol / crypto / parameters / ECDomainParameters.cs
#if !BESTHTTP_DISABLE_ALTERNATE_SSL && (!UNITY_WEBGL || UNITY_EDITOR)
#pragma warning disable
using System;

using BestHTTP.SecureProtocol.Org.BouncyCastle.Math;
using BestHTTP.SecureProtocol.Org.BouncyCastle.Math.EC;
using BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities;

namespace BestHTTP.SecureProtocol.Org.BouncyCastle.Crypto.Parameters
{
    public class ECDomainParameters
    {
        private readonly ECCurve     curve;
        private readonly byte[] seed;
        private readonly ECPoint g;
        private readonly BigInteger n;
        private readonly BigInteger h;

        private BigInteger hInv;

        public ECDomainParameters(
            ECCurve     curve,
            ECPoint     g,
            BigInteger  n)
            : this(curve, g, n, BigInteger.One, null)
        {
        }

        public ECDomainParameters(
            ECCurve     curve,
            ECPoint     g,
            BigInteger  n,
            BigInteger  h)
            : this(curve, g, n, h, null)
        {
        }

        public ECDomainParameters(
            ECCurve     curve,
            ECPoint     g,
            BigInteger  n,
            BigInteger  h,
            byte[]      seed)
        {
            if (curve == null)
                throw new ArgumentNullException("curve");
            if (g == null)
                throw new ArgumentNullException("g");
            if (n == null)
                throw new ArgumentNullException("n");
            // we can't check for h == null here as h is optional in X9.62 as it is not required for ECDSA

            this.curve = curve;
            this.g = ValidatePublicPoint(curve, g);
            this.n = n;
            this.h = h;
            this.seed = Arrays.Clone(seed);
        }

        public ECCurve Curve
        {
            get { return curve; }
        }

        public ECPoint G
        {
            get { return g; }
        }

        public BigInteger N
        {
            get { return n; }
        }

        public BigInteger H
        {
            get { return h; }
        }

        public BigInteger HInv
        {
            get
            {
                lock (this)
                {
                    if (hInv == null)
                    {
                        hInv = h.ModInverse(n);
                    }
                    return hInv;
                }
            }
        }

        public byte[] GetSeed()
        {
            return Arrays.Clone(seed);
        }

        public override bool Equals(
            object obj)
        {
            if (obj == this)
                return true;

            ECDomainParameters other = obj as ECDomainParameters;

            if (other == null)
                return false;

            return Equals(other);
        }

        protected virtual bool Equals(
            ECDomainParameters other)
        {
            return curve.Equals(other.curve)
                &&	g.Equals(other.g)
                &&	n.Equals(other.n);
        }

        public override int GetHashCode()
        {
            //return Arrays.GetHashCode(new object[]{ curve, g, n });
            int hc = 4;
            hc *= 257;
            hc ^= curve.GetHashCode();
            hc *= 257;
            hc ^= g.GetHashCode();
            hc *= 257;
            hc ^= n.GetHashCode();
            return hc;
        }

        public BigInteger ValidatePrivateScalar(BigInteger d)
        {
            if (null == d)
                throw new ArgumentNullException("d", "Scalar cannot be null");

            if (d.CompareTo(BigInteger.One) < 0 || (d.CompareTo(N) >= 0))
                throw new ArgumentException("Scalar is not in the interval [1, n - 1]", "d");

            return d;
        }

        public ECPoint ValidatePublicPoint(ECPoint q)
        {
            return ValidatePublicPoint(Curve, q);
        }

        internal static ECPoint ValidatePublicPoint(ECCurve c, ECPoint q)
        {
            if (null == q)
                throw new ArgumentNullException("q", "Point cannot be null");

            q = ECAlgorithms.ImportPoint(c, q).Normalize();

            if (q.IsInfinity)
                throw new ArgumentException("Point at infinity", "q");

            if (!q.IsValid())
                throw new ArgumentException("Point not on curve", "q");

            return q;
        }
    }
}
#pragma warning restore
#endif